Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: handle empty paths #19

Merged
merged 2 commits into from
Jul 14, 2022
Merged

Conversation

adriangb
Copy link
Contributor

@adriangb adriangb commented Jul 14, 2022

I needed this so that I can compose routers, something like:

Router(
  {
    "/users": Router(
      {
        "": users_endpoint,
        "/:name": user_endpoint,
       }
    )
)

Basically the "value" can be an endpoint or another router, but currently the router assumes/forces the path to start with "/" which in this example means it's impossible to match "/users".

I did this in a TDD style because I'm unfamiliar with the codebase and barely know Rust: I wrote the test and then hacked until I got it to work. I'm still pretty new to rust so please feel free to rip my work apart and apologies if this is missing some obvious edge case that I didn't pick up from the code.

@fundon
Copy link
Member

fundon commented Jul 14, 2022

Currently, path-tree with strict mode. /users and /users/ as different.

In my practice, when nested routes,

/users is a prefix path,
/users is a route.
/users/:name is another route.

We can do something about the path.

    pub fn nest<S>(self, path: S, router: Self) -> Self
    where
        S: AsRef<str>,
    {
        let mut path = path.as_ref().to_string();
        if !path.ends_with('/') {
            path.push('/');
        }

        match router.routes {
            Some(routes) => routes.into_iter().fold(self, |router, (mut sp, route)| {
                let is_empty = sp.is_empty();
                sp = path.clone() + &sp;
                if is_empty {
                    sp = sp.trim_end_matches('/').to_string();
                }
                router.route(sp, route)
            }),
            None => self,
        }
    }
router.nest("/users", Router { routes: vec![("", handler), (":name", handler)] })  

The above code is from Viz, I'm refactoring it and haven't opened it up yet.

@adriangb
Copy link
Contributor Author

adriangb commented Jul 14, 2022

Currently, path-tree with strict mode. /users and /users/ as different.

This would preserve that. As far as I can tell the only thing this changes is allowing "" as a path, which like you say would be different from "/".

router.nest("/users", Router { routes: vec![("", handler), (":name", handler)] })

Hmm I don't particularly like the idea of special casing this, I'd prefer this to be more general and done at a higher level of abstraction. Basically:

  1. Define a route/application trait/API. In the Python world, that's ASGI.
  2. Make a type/class that wraps the low level router (PathTree) and implements that interface (Python example)
  3. Routers match a path, strip it from the remaining path and forward the request to a route. If that route happens to be a router, it can do further matching based on the remaining path. Otherwise if it is an endpoint, it can check the HTTP method or call into the user's code.

Now this does mean a small performance hit since you're splitting up the tree into two, but since it naturally maintains a tree-like structure if the path prefixes are long it could even be a performance win (I think) over grabbing the nested router's routes and adding them to the outermost router's tree. It also provides more flexibility, composability and decoupling.

Ignoring a bunch of details, here's my super clunky and wrong Rust version of the idea

#[derive(Clone, Debug)]
pub struct Request {
    raw_path: String,
    remaining_path: String,
}

pub trait Route {
    fn handle(&self, request: Request) -> ();
}


struct Router {
    tree: PathTree<Box<dyn Route>>
}

impl Route for Router {
    fn handle(&self, request: Request) -> () {
        match self.tree.find(&request.remaining_path.to_owned()) {
            Some((route, params)) => {
                route.handle(
                    Request {
                        raw_path: request.raw_path,
                        remaining_path: request.remaining_path.strip_prefix("the matched path").unwrap().to_string()
                    }
                )
            },
            None => panic!("404")
        }
    }
}

@fundon fundon merged commit 1b04916 into viz-rs:master Jul 14, 2022
@adriangb adriangb deleted the handle-empty-paths branch July 14, 2022 22:44
@adriangb
Copy link
Contributor Author

Thank you for reviewing and merging! Do you plan to cut a release with this change?

@fundon
Copy link
Member

fundon commented Jul 15, 2022

Bumped in v0.3.1.

@fundon
Copy link
Member

fundon commented Jul 16, 2022

@adriangb hi, I opened viz-v0.2.0-alpha. If you are interested, you can take a look at the router implementation.

Currently, I am currently writing docs.

@adriangb
Copy link
Contributor Author

@fundon this seems to have been regressed when you did the rewrite. Do you think the new version could also support this behavior?

@fundon
Copy link
Member

fundon commented Sep 23, 2022

@fundon this seems to have been regressed when you did the rewrite. Do you think the new version could also support this behavior?

Currently, not accept empty string, I thing we can remove the limit.

@adriangb
Copy link
Contributor Author

Yeah that would be great! I gave it a bit of try and couldn't immediately figure out how to get it working without breaking other things, but that's probably just because of how bad my Rust is 😅

@fundon fundon mentioned this pull request Sep 24, 2022
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants